/*****************************************************************************
 *   Copyright(C)2009-2022 by VSF Team                                       *
 *                                                                           *
 *  Licensed under the Apache License, Version 2.0 (the "License");          *
 *  you may not use this file except in compliance with the License.         *
 *  You may obtain a copy of the License at                                  *
 *                                                                           *
 *     http://www.apache.org/licenses/LICENSE-2.0                            *
 *                                                                           *
 *  Unless required by applicable law or agreed to in writing, software      *
 *  distributed under the License is distributed on an "AS IS" BASIS,        *
 *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. *
 *  See the License for the specific language governing permissions and      *
 *  limitations under the License.                                           *
 *                                                                           *
 ****************************************************************************/

/*============================ INCLUDES ======================================*/

#include "hal/driver/common/template/vsf_template_hal_driver.h"

/*============================ MACROS ========================================*/
#if VSF_HAL_USE_GPIO == ENABLED

#ifndef VSF_GPIO_CFG_IMPLEMENT_OP
#   define VSF_GPIO_CFG_IMPLEMENT_OP                    DISABLED
#endif

#if VSF_GPIO_CFG_MULTI_CLASS == DISABLED
#   define VSF_GPIO_OP
#else
#   define ____VSF_GPIO_OP(__p) .vsf_gpio.op = & __p ## _gpio_op,
#   define __VSF_GPIO_OP(__p)   ____VSF_GPIO_OP(__p)
#   define VSF_GPIO_OP          __VSF_GPIO_OP(VSF_GPIO_CFG_INSTANCE_PREFIX)
#endif

#ifndef VSF_GPIO_CFG_INSTANCE_PREFIX
#   define VSF_GPIO_CFG_INSTANCE_PREFIX                 vsf_hw
#endif
#define __GPIO_PREFIX                                   VSF_GPIO_CFG_INSTANCE_PREFIX

#ifndef VSF_GPIO_CFG_REIMPLEMENT_SET_INPUT
#   define VSF_GPIO_CFG_REIMPLEMENT_SET_INPUT           DISABLED
#endif

#ifndef VSF_GPIO_CFG_REIMPLEMENT_SET_OUTPUT
#   define VSF_GPIO_CFG_REIMPLEMENT_SET_OUTPUT          DISABLED
#endif

#ifndef VSF_GPIO_CFG_REIMPLEMENT_SWITCH_DIRECTION
#   define VSF_GPIO_CFG_REIMPLEMENT_SWITCH_DIRECTION    DISABLED
#endif

#ifndef VSF_GPIO_CFG_REIMPLEMENT_SET
#   define VSF_GPIO_CFG_REIMPLEMENT_SET                 DISABLED
#endif

#ifndef VSF_GPIO_CFG_REIMPLEMENT_CLEAR
#   define VSF_GPIO_CFG_REIMPLEMENT_CLEAR               DISABLED
#endif

#ifndef VSF_GPIO_CFG_REIMPLEMENT_OUTPUT_AND_SET
#   define VSF_GPIO_CFG_REIMPLEMENT_OUTPUT_AND_SET      DISABLED
#endif

#ifndef VSF_GPIO_CFG_REIMPLEMENT_OUTPUT_AND_CLEAR
#   define VSF_GPIO_CFG_REIMPLEMENT_OUTPUT_AND_CLEAR    DISABLED
#endif

/*============================ MACROFIED FUNCTIONS ===========================*/

#define VSF_GPIO_FN_CONFIG_PIN        VSF_MCONNECT(__GPIO_PREFIX, _gpio_config_pin)
#define VSF_GPIO_FN_SET_DIRECTION     VSF_MCONNECT(__GPIO_PREFIX, _gpio_set_direction)
#define VSF_GPIO_FN_GET_DIRECTION     VSF_MCONNECT(__GPIO_PREFIX, _gpio_get_direction)
#define VSF_GPIO_FN_SET_INPUT         VSF_MCONNECT(__GPIO_PREFIX, _gpio_set_input)
#define VSF_GPIO_FN_SET_OUTPUT        VSF_MCONNECT(__GPIO_PREFIX, _gpio_set_output)
#define VSF_GPIO_FN_SWITCH_DIRECTION  VSF_MCONNECT(__GPIO_PREFIX, _gpio_switch_direction)
#define VSF_GPIO_FN_READ              VSF_MCONNECT(__GPIO_PREFIX, _gpio_read)
#define VSF_GPIO_FN_WRITE             VSF_MCONNECT(__GPIO_PREFIX, _gpio_write)
#define VSF_GPIO_FN_SET               VSF_MCONNECT(__GPIO_PREFIX, _gpio_set)
#define VSF_GPIO_FN_CLEAR             VSF_MCONNECT(__GPIO_PREFIX, _gpio_clear)
#define VSF_GPIO_FN_TOGGLE            VSF_MCONNECT(__GPIO_PREFIX, _gpio_toggle)
#define VSF_GPIO_FN_OUTPUT_AND_SET    VSF_MCONNECT(__GPIO_PREFIX, _gpio_output_and_set)
#define VSF_GPIO_FN_OUTPUT_AND_CLEAR  VSF_MCONNECT(__GPIO_PREFIX, _gpio_output_and_clear)

#define __VSF_GPIO_APIS(__prefix_name)                                                                                         \
    __VSF_GPIO_API(__prefix_name, void, config_pin, vsf_gpio_t *gpio_ptr, uint32_t pin_mask, uint_fast32_t feature)            \
    __VSF_GPIO_API(__prefix_name, void,     set_direction,    vsf_gpio_t *gpio_ptr, uint32_t direction_mask, uint32_t pin_mask)\
    __VSF_GPIO_API(__prefix_name, uint32_t, get_direction,    vsf_gpio_t *gpio_ptr, uint32_t pin_mask)                         \
    __VSF_GPIO_API(__prefix_name, void,     set_input,        vsf_gpio_t *gpio_ptr, uint32_t pin_mask)                         \
    __VSF_GPIO_API(__prefix_name, void,     set_output,       vsf_gpio_t *gpio_ptr, uint32_t pin_mask)                         \
    __VSF_GPIO_API(__prefix_name, void,     switch_direction, vsf_gpio_t *gpio_ptr, uint32_t pin_mask)                         \
    __VSF_GPIO_API(__prefix_name, uint32_t, read,             vsf_gpio_t *gpio_ptr)                                            \
    __VSF_GPIO_API(__prefix_name, void,     write,            vsf_gpio_t *gpio_ptr, uint32_t value, uint32_t pin_mask)         \
    __VSF_GPIO_API(__prefix_name, void,     set,              vsf_gpio_t *gpio_ptr, uint32_t pin_mask)                         \
    __VSF_GPIO_API(__prefix_name, void,     clear,            vsf_gpio_t *gpio_ptr, uint32_t pin_mask)                         \
    __VSF_GPIO_API(__prefix_name, void,     toggle,           vsf_gpio_t *gpio_ptr, uint32_t pin_mask)                         \
    __VSF_GPIO_API(__prefix_name, void,     output_and_set,   vsf_gpio_t *gpio_ptr, uint32_t pin_mask)                         \
    __VSF_GPIO_API(__prefix_name, void,     output_and_clear, vsf_gpio_t *gpio_ptr, uint32_t pin_mask)

/*============================ PROTOTYPES ====================================*/

#undef   __VSF_GPIO_API
#define  __VSF_GPIO_API(__prefix_name, __return, __name, ...)                   \
    VSF_TEMPLATE_HAL_API_EXTERN(__prefix_name, _gpio_, __return, __name, __VA_ARGS__)

__VSF_GPIO_APIS(VSF_GPIO_CFG_INSTANCE_PREFIX)

/*============================ LOCAL VARIABLES ===============================*/

#if VSF_GPIO_CFG_MULTI_CLASS == ENABLED && VSF_GPIO_CFG_IMPLEMENT_OP == ENABLED
static const vsf_gpio_op_t vsf_hw_gpio_op = {

#undef   __VSF_GPIO_API
#define  __VSF_GPIO_API(__prefix_name, __return, __name, ...)                   \
    VSF_TEMPLATE_HAL_API_OP(__prefix_name, _gpio_, __return, __name, __VA_ARGS__)

    __VSF_GPIO_APIS(VSF_GPIO_CFG_INSTANCE_PREFIX)
};
#endif

/*============================ MACROFIED FUNCTIONS ===========================*/
/*============================ IMPLEMENTATION ================================*/

#if VSF_GPIO_CFG_REIMPLEMENT_SET_INPUT == DISALBED
void VSF_GPIO_FN_SET_INPUT(vsf_gpio_t *gpio_ptr, uint32_t pin_mask)
{
    VSF_HAL_ASSERT(NULL != gpio_ptr);

    vsf_gpio_set_direction(gpio_ptr, 0, pin_mask);
}
#endif

#if VSF_GPIO_CFG_REIMPLEMENT_SET_OUTPUT == DISALBED
void VSF_GPIO_FN_SET_OUTPUT(vsf_gpio_t *gpio_ptr, uint32_t pin_mask)
{
    VSF_HAL_ASSERT(NULL != gpio_ptr);

    vsf_gpio_set_direction(gpio_ptr, pin_mask, pin_mask);
}
#endif

#if VSF_GPIO_CFG_REIMPLEMENT_SWITCH_DIRECTION == DISALBED
void VSF_GPIO_FN_SWITCH_DIRECTION(vsf_gpio_t *gpio_ptr, uint32_t pin_mask)
{
    VSF_HAL_ASSERT(NULL != gpio_ptr);

    uint32_t ret = ~vsf_gpio_get_direction(gpio_ptr, pin_mask);
    vsf_gpio_set_direction(gpio_ptr, ret, pin_mask);
}
#endif

#if VSF_GPIO_CFG_REIMPLEMENT_SET == DISALBED
void VSF_GPIO_FN_SET(vsf_gpio_t *gpio_ptr, uint32_t pin_mask)
{
    VSF_HAL_ASSERT(NULL != gpio_ptr);

    vsf_gpio_write(gpio_ptr, pin_mask, pin_mask);
}
#endif

#if VSF_GPIO_CFG_REIMPLEMENT_CLEAR == DISALBED
void VSF_GPIO_FN_CLEAR(vsf_gpio_t *gpio_ptr, uint32_t pin_mask)
{
    VSF_HAL_ASSERT(NULL != gpio_ptr);

    vsf_gpio_write(gpio_ptr, 0, pin_mask);
}
#endif

#if VSF_GPIO_CFG_REIMPLEMENT_OUTPUT_AND_SET == DISALBED
void VSF_GPIO_FN_OUTPUT_AND_SET(vsf_gpio_t *gpio_ptr, uint32_t pin_mask)
{
    VSF_HAL_ASSERT(NULL != gpio_ptr);

    vsf_gpio_set(gpio_ptr, pin_mask);
    vsf_gpio_set_output(gpio_ptr, pin_mask);
}
#endif

#if VSF_GPIO_CFG_REIMPLEMENT_OUTPUT_AND_CLEAR == DISALBED
void VSF_GPIO_FN_OUTPUT_AND_CLEAR(vsf_gpio_t *gpio_ptr, uint32_t pin_mask)
{
    VSF_HAL_ASSERT(NULL != gpio_ptr);

    vsf_gpio_clear(gpio_ptr, pin_mask);
    vsf_gpio_set_output(gpio_ptr, pin_mask);
}
#endif

/*============================ GLOBAL VARIABLES ==============================*/

#ifdef VSF_GPIO_CFG_TEMPLATE_COUNT
#   ifndef VSF_GPIO_CFG_TEMPLATE_MASK
#       define VSF_GPIO_CFG_TEMPLATE_MASK     ((1ul << VSF_GPIO_CFG_TEMPLATE_COUNT) - 1)
#   endif

#   if VSF_GPIO_CFG_TEMPLATE_MASK & (1 << 0)
        VSF_GPIO_CFG_IMP_LV0(0, NULL)
#   endif
#   if VSF_GPIO_CFG_TEMPLATE_MASK & (1 << 1)
        VSF_GPIO_CFG_IMP_LV0(1, NULL)
#   endif
#   if VSF_GPIO_CFG_TEMPLATE_MASK & (1 << 2)
        VSF_GPIO_CFG_IMP_LV0(2, NULL)
#   endif
#   if VSF_GPIO_CFG_TEMPLATE_MASK & (1 << 3)
        VSF_GPIO_CFG_IMP_LV0(3, NULL)
#   endif
#   if VSF_GPIO_CFG_TEMPLATE_MASK & (1 << 4)
        VSF_GPIO_CFG_IMP_LV0(4, NULL)
#   endif
#   if VSF_GPIO_CFG_TEMPLATE_MASK & (1 << 5)
        VSF_GPIO_CFG_IMP_LV0(5, NULL)
#   endif
#   if VSF_GPIO_CFG_TEMPLATE_MASK & (1 << 6)
        VSF_GPIO_CFG_IMP_LV0(6, NULL)
#   endif
#   if VSF_GPIO_CFG_TEMPLATE_MASK & (1 << 7)
        VSF_GPIO_CFG_IMP_LV0(7, NULL)
#   endif

#   undef VSF_GPIO_CFG_TEMPLATE_COUNT
#   undef VSF_GPIO_CFG_TEMPLATE_MASK
#   undef VSF_GPIO_CFG_IMP_LV0
#endif

#endif // VSF_HAL_USE_GPIO == ENABLED
